package cn.tedu.csmall.passport.config;

import cn.tedu.csmall.commons.web.JsonResult;
import cn.tedu.csmall.commons.web.ServiceCode;
import cn.tedu.csmall.passport.filter.JwtAuthorizationFilter;

import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.HttpMethod;
import org.springframework.security.authentication.AuthenticationManager;
import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;
import org.springframework.security.config.http.SessionCreationPolicy;
import org.springframework.security.core.AuthenticationException;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
import org.springframework.security.crypto.password.PasswordEncoder;
import org.springframework.security.web.AuthenticationEntryPoint;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;


@Slf4j
@Configuration
@EnableGlobalMethodSecurity(prePostEnabled = true) //开启全局的基于方法的安全检查，即：在方法上添加注解来检查权限
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {

    @Bean
    public PasswordEncoder passwordEncoder() {
//        return NoOpPasswordEncoder.getInstance();
        return new BCryptPasswordEncoder();
        //使用BCrypt加密算法
    }

    @Bean
    @Override
    public AuthenticationManager authenticationManagerBean() throws Exception {
        return super.authenticationManagerBean();
    }

    @Autowired
    private JwtAuthorizationFilter jwtAuthorizationFilter;

    @Override
    protected void configure(HttpSecurity http) throws Exception {

        //开启Spring Security自带的CorsFilter，以解决跨域问题
        //解决复杂请求的跨越问题
        http.cors();

        //配置Spring Security框架使用创建Session的策略
        //STATELESS:无状态的，完全不使用Session
        //NEVER：从不主动创建Session，但是，当Session已经被创建后，仍会正常使用
        http.sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS);


        //将JWT过滤器添加到Spring Security框架的某些过滤器之前
        http.addFilterBefore(jwtAuthorizationFilter, UsernamePasswordAuthenticationFilter.class);


        //处理未通过认证时访问受保护的资源时拒绝访问
        http.exceptionHandling().authenticationEntryPoint(new AuthenticationEntryPoint() {
            @Override
            public void commence(HttpServletRequest request, HttpServletResponse response, AuthenticationException e) throws IOException, ServletException {
                response.setContentType("application/json; chatset=utf-8");
                String message = "您当前未登录，请先登录！";
                log.warn(message);

                JsonResult jsonResult = JsonResult.fail(ServiceCode.ERR_UNAUTHORIZED,message);
                String jsonString = JSON.toJSONString(jsonResult);

                response.setContentType("application/json; charset=utf-8");
                PrintWriter printWriter = response.getWriter();
                printWriter.println(jsonString);
                printWriter.close();

            }
        });



        //解决403问题
        //禁用防止伪造的跨域攻击的防御机制导致POST请求不可用
        http.csrf().disable();



        //白名单
        // 使用1个星号，可以通配此层级的任何资源，
        // 例如：
        // /admin/*，可以匹配：/admin/add-new、/admin/list，但不可以匹配：/admin/password/change
        // 使用2个连续的星可以，可以通配若干层级的资源，
        // 例如：
        // /admin/**，可以匹配：/admin/add-new、/admin/password/change

        String[] urls = {
                "/doc.html",
                "/**/*.css",
                "/**/*.js",
                "/swagger-resources",
                "/v2/api-docs",
                "/admins/login"


        };

        //配置授权访问
        //注意：以下授权访问的配置，是遵循“第一匹配原则“的即以最先匹配的规则为准
        //例如：anyRequest()是匹配任何请求，通常，应该配置在最后，表示除了以上配置过的以外的所有配置
        //所以，在开发实践中，应该将更具体的请求配置在靠前的位置，将更庞统的请求配置在靠后的位置
        http.authorizeRequests()//开始对请求进行授权



                .mvcMatchers(urls)//匹配某些请求
                .permitAll() //许可，即不需要通过认证就可以访问
                .anyRequest()     //任何请求
                .authenticated(); //已经完成认证的


        //super.configure(http); 不要保留调用父级同名方法的代码，不要保留！！！！！


        //如果调用以下方法，当Security认为需要通过认证，但实际未通过认证时，就会跳转到
        //登录页面

        //如果未调用以下方法，将会响应403错误
        http.formLogin();
    }
}
